home *** CD-ROM | disk | FTP | other *** search
/ Night Owl 7 / Night Owl Shareware (NOPV7)(Night Owl Publisher Inc.)(1992).bin / 038a / bash1_12.arj / BASH1-12.TAR / bash-1.12 / builtins / set.def < prev    next >
Text File  |  1991-12-29  |  13KB  |  486 lines

  1. This file is set.def, from which is created set.c.
  2. It implements the "set" and "unset" builtins in Bash.
  3.  
  4. Copyright (C) 1987, 1989, 1991 Free Software Foundation, Inc.
  5.  
  6. This file is part of GNU Bash, the Bourne Again SHell.
  7.  
  8. Bash is free software; you can redistribute it and/or modify it under
  9. the terms of the GNU General Public License as published by the Free
  10. Software Foundation; either version 1, or (at your option) any later
  11. version.
  12.  
  13. Bash is distributed in the hope that it will be useful, but WITHOUT ANY
  14. WARRANTY; without even the implied warranty of MERCHANTABILITY or
  15. FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  16. for more details.
  17.  
  18. You should have received a copy of the GNU General Public License along
  19. with Bash; see the file COPYING.  If not, write to the Free Software
  20. Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
  21.  
  22. $PRODUCES set.c
  23.  
  24. #include <stdio.h>
  25. #include "../shell.h"
  26. #include "../flags.h"
  27.  
  28. extern int interactive;
  29.  
  30. $BUILTIN set
  31. $FUNCTION set_builtin
  32. $SHORT_DOC set [-abefhknotuvxldH] [arg ...]
  33.     -a  Mark variables which are modified or created for export
  34.     -b  Notify of job termination immediately
  35.     -e  Exit immediately if a command exits with a non-zero status
  36.     -f  Disable file name generation (globbing)
  37.     -h  Locate and remember function commands as functions are
  38.         defined.  Function commands are normally looked up when
  39.         the function is executed
  40.     -k  All keyword arguments are placed in the environment for a
  41.         command, not just those that precede the command name
  42.     -m  Job control is enabled
  43.     -n  Read commands but do not execute them
  44.     -o option-name
  45.         Set the variable corresponding to option-name:
  46.             allexport    same as -a
  47.             braceexpand  the shell will perform brace expansion
  48.             emacs        use an emacs-style line editing interface
  49.             errexit      same as -e
  50.             histexpand   same as -H
  51.             ignoreeof    the shell will not exit upon reading EOF
  52.             monitor      same as -m
  53.             noclobber    disallow redirection to existing files
  54.             noexec       same as -n
  55.             noglob       same as -f
  56.             nohash       same as -d
  57.             notify       save as -b
  58.             nounset      same as -u
  59.             verbose      same as -v
  60.             vi           use a vi-style line editing interface
  61.             xtrace       same as -x
  62.     -t  Exit after reading and executing one command
  63.     -u  Treat unset variables as an error when substituting
  64.     -v  Print shell input lines as they are read
  65.     -x  Print commands and their arguments as they are executed
  66.     -l  Save and restore the binding of the NAME in a FOR command.
  67.     -d  Disable the hashing of commands that are looked up for execution.
  68.         Normally, commands are remembered in a hash table, and once
  69.         found, do not have to be looked up again
  70.     -H  Enable ! style history substitution.  This flag is on
  71.         by default.
  72.     -C  If set, disallow existing regular files to be overwritten
  73.         by redirection of output.
  74.  
  75. Using + rather than - causes these flags to be turned off.  The
  76. flags can also be used upon invocation of the shell.  The current
  77. set of flags may be found in $-.  The remaining ARGs are positional
  78. parameters and are assigned, in order, to $1, $2, .. $9.  If no
  79. ARGs are given, all shell variables are printed.
  80. $END
  81.  
  82. /* An a-list used to match long options for set -o to the corresponding
  83.    option letter. */
  84. struct {
  85.   char *name;
  86.   int letter;
  87. } o_options[] = {
  88.   { "allexport",  'a' },
  89.   { "errexit",      'e' },
  90.   { "histexpand", 'H' },
  91.   { "monitor",      'm' },
  92.   { "noexec",      'n' },
  93.   { "noglob",      'f' },
  94.   { "nohash",      'd' },
  95. #if defined (JOB_CONTROL)
  96.   { "notify",      'b' },
  97. #endif /* JOB_CONTROL */
  98.   {"nounset",      'u' },
  99.   {"verbose",      'v' },
  100.   {"xtrace",      'x' },
  101.   {(char *)NULL, 0},
  102. };
  103.  
  104. static void
  105. list_long_opts ()
  106. {
  107.   register int    i;
  108.   char *on = "on", *off = "off";
  109.   extern int noclobber, no_brace_expansion;
  110.   extern SHELL_VAR *find_variable ();
  111.  
  112. #if defined (READLINE)
  113.   extern int rl_editing_mode, no_line_editing;
  114. #endif /* READLINE */
  115.  
  116.   printf ("%-15s\t%s\n", "braceexpand", (no_brace_expansion == 0) ? on : off);
  117.   printf ("%-15s\t%s\n", "noclobber", (noclobber == 1) ? on : off);
  118.  
  119.   if (find_variable ("ignoreeof") || find_variable ("IGNOREEOF"))
  120.     printf ("%-15s\t%s\n", "ignoreeof", on);
  121.   else
  122.     printf ("%-15s\t%s\n", "ignoreeof", off);
  123.  
  124. #if defined (READLINE)
  125.   if (no_line_editing)
  126.     {
  127.       printf ("%-15s\toff\n", "emacs");
  128.       printf ("%-15s\toff\n", "vi");
  129.     }
  130.   else
  131.     {
  132.       /* Magic.  This code `knows' how readline handles rl_editing_mode. */
  133.       printf ("%-15s\t%s\n", "emacs", (rl_editing_mode == 1) ? on : off);
  134.       printf ("%-15s\t%s\n", "vi", (rl_editing_mode == 0) ? on : off);
  135.     }
  136. #endif /* READLINE */
  137.  
  138.   for (i = 0; o_options[i].name; i++)
  139.     {
  140.       int *on_or_off, zero = 0;
  141.       char name[2];
  142.  
  143.       name[0] = o_options[i].letter;
  144.       name[1] = '\0';
  145.       on_or_off = find_flag (name);
  146.       if (on_or_off == (int *)FLAG_ERROR)
  147.     on_or_off = &zero;
  148.       printf ("%-15s\t%s\n", o_options[i].name, (*on_or_off == 1) ? on : off);
  149.     }
  150. }
  151.  
  152. set_minus_o_option (on_or_off, option_name)
  153.      int on_or_off;
  154.      char *option_name;
  155. {
  156.   extern int no_brace_expansion;
  157.   int option_char = -1;
  158.  
  159.   if (strcmp (option_name, "braceexpand") == 0)
  160.     {
  161.       if (on_or_off == FLAG_ON)
  162.     no_brace_expansion = 0;
  163.       else
  164.     no_brace_expansion = 1;
  165.     }
  166.   else if (strcmp (option_name, "noclobber") == 0)
  167.     {
  168.       if (on_or_off == FLAG_ON)
  169.     bind_variable ("noclobber", "");
  170.       else
  171.     unbind_variable ("noclobber");
  172.       stupidly_hack_special_variables ("noclobber");
  173.     }
  174.   else if (strcmp (option_name, "ignoreeof") == 0)
  175.     {
  176.       unbind_variable ("ignoreeof");
  177.       unbind_variable ("IGNOREEOF");
  178.       if (on_or_off == FLAG_ON)
  179.     bind_variable ("IGNOREEOF", "10");
  180.       stupidly_hack_special_variables ("IGNOREEOF");
  181.     }
  182.  
  183. #if defined (READLINE)
  184.   else if ((strcmp (option_name, "emacs") == 0) ||
  185.        (strcmp (option_name, "vi") == 0))
  186.     {
  187.       extern int no_line_editing;
  188.  
  189.       if (on_or_off == FLAG_ON)
  190.     {
  191.       rl_variable_bind ("editing-mode", option_name);
  192.  
  193.       if (interactive)
  194.         with_input_from_stdin ();
  195.       no_line_editing = 0;
  196.     }
  197.       else
  198.     {
  199.       extern int rl_editing_mode;
  200.       int isemacs = (rl_editing_mode == 1);
  201.       if ((isemacs && strcmp (option_name, "emacs") == 0) ||
  202.           (!isemacs && strcmp (option_name, "vi") == 0))
  203.         {
  204.           if (interactive)
  205.         with_input_from_stream (stdin, "stdin");
  206.           no_line_editing = 1;
  207.         }
  208.       else
  209.         builtin_error ("not in %s editing mode",
  210.                 option_name);
  211.     }
  212.     }
  213. #endif /* READLINE */
  214.   else
  215.     {
  216.       register int i;
  217.       for (i = 0; o_options[i].name; i++)
  218.     {
  219.       if (strcmp (option_name, o_options[i].name) == 0)
  220.         {
  221.           option_char = o_options[i].letter;
  222.           break;
  223.         }
  224.     }
  225.       if (option_char == -1)
  226.     {
  227.       builtin_error ("%s: unknown option name", option_name);
  228.       return (EXECUTION_FAILURE);
  229.     }
  230.       if (change_flag_char (option_char, on_or_off) == FLAG_ERROR)
  231.     {
  232.       bad_option (option_name);
  233.       return (EXECUTION_FAILURE);
  234.     }
  235.     }
  236.     return (EXECUTION_SUCCESS);
  237. }
  238.  
  239. /* Set some flags from the word values in the input list.  If LIST is empty,
  240.    then print out the values of the variables instead.  If LIST contains
  241.    non-flags, then set $1 - $9 to the successive words of LIST. */
  242. set_builtin (list)
  243.      WORD_LIST *list;
  244. {
  245.   int on_or_off, flag_name, force_assignment = 0;
  246.  
  247.   if (!list)
  248.     {
  249.       SHELL_VAR **vars;
  250.  
  251.       vars = all_shell_variables ();
  252.       if (vars)
  253.     {
  254.       print_var_list (vars);
  255.       free (vars);
  256.     }
  257.  
  258.       vars = all_shell_functions ();
  259.       if (vars)
  260.     {
  261.       print_var_list (vars);
  262.       free (vars);
  263.     }
  264.  
  265.       return (EXECUTION_SUCCESS);
  266.     }
  267.  
  268.   /* Check validity of flag arguments. */
  269.   if (*list->word->word == '-' || *list->word->word == '+')
  270.     {
  271.       register char *arg;
  272.       WORD_LIST *save_list = list;
  273.  
  274.       while (list && (arg = list->word->word))
  275.     {
  276.       char s[2];
  277.  
  278.       if (arg[0] != '-' && arg[0] != '+')
  279.         break;
  280.  
  281.       /* `-' or `--' signifies end of flag arguments. */
  282.       if (arg[0] == '-' &&
  283.           (!arg[1] || (arg[1] == '-' && !arg[2])))
  284.         break;
  285.  
  286.       s[1] = '\0';
  287.  
  288.       while (s[0] = *++arg)
  289.         {
  290.           if (find_flag (s) == (int *)FLAG_ERROR && s[0] != 'o')
  291.         {
  292.           bad_option (s);
  293.           return (EXECUTION_FAILURE);
  294.         }
  295.         }
  296.       list = list->next;
  297.     }
  298.       list = save_list;
  299.     }
  300.  
  301.   /* Do the set command.  While the list consists of words starting with
  302.      '-' or '+' treat them as flags, otherwise, start assigning them to
  303.      $1 ... $n. */
  304.   while (list)
  305.     {
  306.       char *string = list->word->word;
  307.  
  308.       /* If the argument is `--' then signal the end of the list and
  309.      remember the remaining arguments. */
  310.       if ((strcmp (string, "--") == 0) || (strcmp (string, "-") == 0))
  311.     {
  312.       list = list->next;
  313.  
  314.       /* `set --' unsets the positional parameters. */
  315.       if (strcmp (string, "--") == 0)
  316.         force_assignment = 1;
  317.  
  318.       /* Until told differently, the old shell behaviour of
  319.          `set - [arg ...]' being equivalent to `set +xv [arg ...]'
  320.          stands.  Posix.2 says the behaviour is marked as obsolescent. */
  321.       else
  322.         {
  323.           change_flag_char ('x', '+');
  324.           change_flag_char ('v', '+');
  325.         }
  326.  
  327.       break;
  328.     }
  329.  
  330.       if ((on_or_off = *string) &&
  331.       (on_or_off == '-' || on_or_off == '+'))
  332.     {
  333.       int i = 1;
  334.       while (flag_name = string[i++])
  335.         {
  336.           if (flag_name == '?')
  337.         {
  338.           /* Print all the possible flags. */
  339.         }
  340.           else if (flag_name == 'o') /* -+o option-name */
  341.         {
  342.           char *option_name;
  343.           WORD_LIST *opt;
  344.  
  345.           opt = list->next;
  346.  
  347.           if (!opt)
  348.             {
  349.               list_long_opts ();
  350.               continue;
  351.             }
  352.  
  353.           option_name = opt->word->word;
  354.  
  355.           if (!option_name || !*option_name || (*option_name == '-'))
  356.             {
  357.               list_long_opts ();
  358.               continue;
  359.             }
  360.           list = list->next; /* Skip over option name. */
  361.  
  362.           if (set_minus_o_option
  363.               (on_or_off, option_name) != EXECUTION_SUCCESS)
  364.             return (EXECUTION_FAILURE);
  365.         }
  366.           else
  367.         {
  368.           if (change_flag_char (flag_name, on_or_off) == FLAG_ERROR)
  369.             {
  370.               char opt[3];
  371.               opt[0] = on_or_off;
  372.               opt[1] = flag_name;
  373.               opt[2] = '\0';
  374.               bad_option (opt);
  375.               return (EXECUTION_FAILURE);
  376.             }
  377.         }
  378.         }
  379.     }
  380.       else
  381.     {
  382.       break;
  383.     }
  384.       list = list->next;
  385.     }
  386.  
  387.   /* Assigning $1 ... $n */
  388.   if (list || force_assignment)
  389.     remember_args (list, 1);
  390.   return (EXECUTION_SUCCESS);
  391. }
  392.  
  393. $BUILTIN unset
  394. $FUNCTION unset_builtin
  395. $SHORT_DOC unset [-f] [-v] [name ...]
  396. For each NAME, remove the corresponding variable or function.  Given
  397. the `-v', unset will only act on variables.  Given the `-f' flag,
  398. unset will only act on functions.  With neither flag, unset first
  399. tries to unset a variable, and if that fails, then tries to unset a
  400. function.  Some variables (such as PATH and IFS) cannot be unset; also
  401. see readonly.
  402. $END
  403.  
  404. unset_builtin (list)
  405.   WORD_LIST *list;
  406. {
  407.   extern char **non_unsettable_vars;
  408.   int unset_function = 0, unset_variable = 0;
  409.   int any_failed = 0;
  410.   char *name;
  411.  
  412.   while (list)
  413.     {
  414.       name = list->word->word;
  415.  
  416.       if (strcmp (name, "-f") == 0)
  417.     {
  418.       list = list->next;
  419.       unset_function++;
  420.       continue;
  421.     }
  422.  
  423.       if (strcmp (name, "-v") == 0)
  424.     {
  425.       list = list->next;
  426.       unset_variable++;
  427.       continue;
  428.     }
  429.       else if (strcmp (name, "--") == 0)
  430.     {
  431.       list = list->next;
  432.       break;
  433.     }
  434.       else if (*name == '-')
  435.     {
  436.       bad_option (name);
  437.       return (EXECUTION_FAILURE);
  438.     }
  439.       else
  440.     break;
  441.     }
  442.  
  443.   if (unset_function && unset_variable)
  444.     {
  445.       builtin_error ("cannot simultaneously unset a function and a variable");
  446.       return (EXECUTION_FAILURE);
  447.     }
  448.  
  449.   while (list)
  450.     {
  451.       name = list->word->word;
  452.  
  453.       if (!unset_function &&
  454.       find_name_in_list (name, non_unsettable_vars) > -1)
  455.     {
  456.       builtin_error ("%s: cannot unset", name);
  457.       any_failed++;
  458.     }
  459.       else
  460.     {
  461.       /* Unless the -f option is supplied, the name refers to a
  462.          variable. */
  463.       int tem = makunbound (name,
  464.           unset_function ? shell_functions : shell_variables);
  465.  
  466.       /* This is what Posix.2 draft 11+ says.  ``If neither -f nor -v
  467.          is specified, the name refers to a variable; if a variable by
  468.          that name does not exist, a function by that name, if any,
  469.          shall be unset.'' */
  470.       if ((tem == -1) && !unset_function && !unset_variable)
  471.         tem = makunbound (name, shell_functions);
  472.  
  473.       if (tem == -1)
  474.         any_failed++;
  475.       else if (!unset_function)
  476.         stupidly_hack_special_variables (name);
  477.     }
  478.       list = list->next;
  479.     }
  480.  
  481.   if (any_failed)
  482.     return (EXECUTION_FAILURE);
  483.   else
  484.     return (EXECUTION_SUCCESS);
  485. }
  486.